home *** CD-ROM | disk | FTP | other *** search
/ CU Amiga Super CD-ROM 15 / CU Amiga Magazine's Super CD-ROM 15 (1997)(EMAP Images)(GB)[!][issue 1997-10].iso / CUCD / Graphics / Ghostscript / source / zdevice.c < prev    next >
C/C++ Source or Header  |  1997-06-18  |  10KB  |  349 lines

  1. /* Copyright (C) 1989, 1996, 1997 Aladdin Enterprises.  All rights reserved.
  2.   
  3.   This file is part of Aladdin Ghostscript.
  4.   
  5.   Aladdin Ghostscript is distributed with NO WARRANTY OF ANY KIND.  No author
  6.   or distributor accepts any responsibility for the consequences of using it,
  7.   or for whether it serves any particular purpose or works at all, unless he
  8.   or she says so in writing.  Refer to the Aladdin Ghostscript Free Public
  9.   License (the "License") for full details.
  10.   
  11.   Every copy of Aladdin Ghostscript must include a copy of the License,
  12.   normally in a plain ASCII text file named PUBLIC.  The License grants you
  13.   the right to copy, modify and redistribute Aladdin Ghostscript, but only
  14.   under certain conditions described in the License.  Among other things, the
  15.   License requires that the copyright notice and this notice be preserved on
  16.   all copies.
  17. */
  18.  
  19. /* zdevice.c */
  20. /* Device-related operators */
  21. #include "string_.h"
  22. #include "ghost.h"
  23. #include "errors.h"
  24. #include "oper.h"
  25. #include "ialloc.h"
  26. #include "idict.h"
  27. #include "igstate.h"
  28. #include "iname.h"
  29. #include "interp.h"
  30. #include "iparam.h"
  31. #include "ivmspace.h"
  32. #include "gsmatrix.h"
  33. #include "gsstate.h"
  34. #include "gxdevice.h"
  35. #include "store.h"
  36.  
  37. /* <device> copydevice <newdevice> */
  38. private int
  39. zcopydevice(register os_ptr op)
  40. {    gx_device *new_dev;
  41.     int code;
  42.  
  43.     check_read_type(*op, t_device);
  44.     code = gs_copydevice(&new_dev, op->value.pdevice, imemory);
  45.     if ( code < 0 )
  46.       return code;
  47.     new_dev->memory = imemory;
  48.     make_tav(op, t_device, icurrent_space | a_all, pdevice, new_dev);
  49.     return 0;
  50. }
  51.  
  52. /* <device> <y> <string> copyscanlines <substring> */
  53. private int
  54. zcopyscanlines(register os_ptr op)
  55. {    os_ptr op1 = op - 1;
  56.     os_ptr op2 = op - 2;
  57.     gx_device *dev;
  58.     int code;
  59.     uint bytes_copied;
  60.  
  61.     check_read_type(*op2, t_device);
  62.     dev = op2->value.pdevice;
  63.     check_type(*op1, t_integer);
  64.     if ( op1->value.intval < 0 || op1->value.intval > dev->height )
  65.       return_error(e_rangecheck);
  66.     check_write_type(*op, t_string);
  67.     code = gs_copyscanlines(dev, (int)op1->value.intval,
  68.                 op->value.bytes, r_size(op), NULL,
  69.                 &bytes_copied);
  70.     if ( code < 0 )
  71.       return_error(e_rangecheck);    /* not a memory device */
  72.     *op2 = *op;
  73.     r_set_size(op2, bytes_copied);
  74.     pop(2);
  75.     return 0;
  76. }
  77.  
  78. /* - currentdevice <device> */
  79. int
  80. zcurrentdevice(register os_ptr op)
  81. {    gx_device *dev = gs_currentdevice(igs);
  82.     gs_ref_memory_t *mem = (gs_ref_memory_t *)dev->memory;
  83.  
  84.     push(1);
  85.     make_tav(op, t_device,
  86.          (mem == 0 ? avm_foreign : imemory_space(mem)) | a_all,
  87.          pdevice, dev);
  88.     return 0;
  89. }
  90.  
  91. /* <device> .devicename <string> */
  92. int
  93. zdevicename(register os_ptr op)
  94. {    const char *dname;
  95.  
  96.     check_read_type(*op, t_device);
  97.     dname = op->value.pdevice->dname;
  98.     make_const_string(op, avm_foreign | a_readonly, strlen(dname),
  99.               (const byte *)dname);
  100.     return 0;
  101. }
  102.  
  103. /* - .doneshowpage - */
  104. private int
  105. zdoneshowpage(register os_ptr op)
  106. {    gx_device *dev = gs_currentdevice(igs);
  107.     gx_device *tdev = (*dev_proc(dev, get_page_device))(dev);
  108.  
  109.     if ( tdev != 0 )
  110.       tdev->ShowpageCount++;
  111.     return 0;
  112. }
  113.  
  114. /* - flushpage - */
  115. int
  116. zflushpage(register os_ptr op)
  117. {    return gs_flushpage(igs);
  118. }
  119.  
  120. /* <int> .getdevice <device> */
  121. private int
  122. zgetdevice(register os_ptr op)
  123. {    const gx_device *dev;
  124.  
  125.     check_type(*op, t_integer);
  126.     if ( op->value.intval != (int)(op->value.intval) )
  127.       return_error(e_rangecheck);    /* won't fit in an int */
  128.     dev = gs_getdevice((int)(op->value.intval));
  129.     if ( dev == 0 )        /* index out of range */
  130.       return_error(e_rangecheck);
  131.     /* Device prototypes are read-only; */
  132.     /* the cast is logically unnecessary. */
  133.     make_tav(op, t_device, avm_foreign | a_readonly, pdevice,
  134.          (gx_device *)dev);
  135.     return 0;
  136. }
  137.  
  138. /* <device> <key_dict|null> .getdeviceparams <mark> <name> <value> ... */
  139. private int
  140. zgetdeviceparams(os_ptr op)
  141. {    ref rkeys;
  142.     gx_device *dev;
  143.     stack_param_list list;
  144.     int code;
  145.     ref *pmark;
  146.  
  147.     check_read_type(op[-1], t_device);
  148.     rkeys = *op;
  149.     dev = op[-1].value.pdevice;
  150.     pop(1);
  151.     stack_param_list_write(&list, &o_stack, &rkeys);
  152.     code = gs_getdeviceparams(dev, (gs_param_list *)&list);
  153.     if ( code < 0 )
  154.       {    /* We have to put back the top argument. */
  155.         if ( list.count > 0 )
  156.           ref_stack_pop(&o_stack, list.count * 2 - 1);
  157.         else
  158.           ref_stack_push(&o_stack, 1);
  159.         *osp = rkeys;
  160.         return code;
  161.       }
  162.     pmark = ref_stack_index(&o_stack, list.count * 2);
  163.     make_mark(pmark);
  164.     return 0;
  165. }
  166.  
  167. /* <matrix> <width> <height> <palette> <word?> makewordimagedevice <device> */
  168. private int
  169. zmakewordimagedevice(register os_ptr op)
  170. {    os_ptr op1 = op - 1;
  171.     gs_matrix imat;
  172.     gx_device *new_dev;
  173.     const byte *colors;
  174.     int colors_size;
  175.     int code;
  176.  
  177.     check_int_leu(op[-3], max_uint >> 1);    /* width */
  178.     check_int_leu(op[-2], max_uint >> 1);    /* height */
  179.     check_type(*op, t_boolean);
  180.     if ( r_has_type(op1, t_null) )    /* true color */
  181.        {    colors = 0;
  182.         colors_size = -24;    /* 24-bit true color */
  183.        }
  184.     else if ( r_has_type(op1, t_integer) )
  185.       {    switch ( op1->value.intval )
  186.           {
  187.           case 16: case 24: case 32: break;
  188.           default: return_error(e_rangecheck);
  189.           }
  190.         colors = 0;
  191.         colors_size = -op1->value.intval;
  192.       }
  193.     else
  194.        {    check_type(*op1, t_string);    /* palette */
  195.         if ( r_size(op1) > 3*256 )
  196.           return_error(e_rangecheck);
  197.         colors = op1->value.bytes;
  198.         colors_size = r_size(op1);
  199.        }
  200.     if ( (code = read_matrix(op - 4, &imat)) < 0 )
  201.         return code;
  202.     /* Everything OK, create device */
  203.     code = gs_makewordimagedevice(&new_dev, &imat,
  204.                       (int)op[-3].value.intval,
  205.                       (int)op[-2].value.intval,
  206.                       colors, colors_size,
  207.                       op->value.boolval, true, imemory);
  208.     if ( code == 0 )
  209.     {    new_dev->memory = imemory;
  210.         make_tav(op - 4, t_device, imemory_space(iimemory) | a_all,
  211.              pdevice, new_dev);
  212.         pop(4);
  213.     }
  214.     return code;
  215. }
  216.  
  217. /* - nulldevice - */
  218. /* Note that nulldevice clears the current pagedevice. */
  219. private int
  220. znulldevice(register os_ptr op)
  221. {    gs_nulldevice(igs);
  222.     clear_pagedevice(istate);
  223.     return 0;
  224. }
  225.  
  226. /* <num_copies> <flush_bool> .outputpage - */
  227. private int
  228. zoutputpage(register os_ptr op)
  229. {    int code;
  230.  
  231.     check_type(op[-1], t_integer);
  232.     check_type(*op, t_boolean);
  233.     code = gs_output_page(igs, (int)op[-1].value.intval,
  234.                   op->value.boolval);
  235.     if ( code < 0 )
  236.       return code;
  237.     pop(2);
  238.     return 0;
  239. }
  240.  
  241. /* <device> <policy_dict|null> <require_all> <mark> <name> <value> ... */
  242. /*    .putdeviceparams */
  243. /*   (on success) <device> <eraseflag> */
  244. /*   (on failure) <device> <policy_dict|null> <require_all> <mark> */
  245. /*     <name1> <error1> ... */
  246. /* For a key that simply was not recognized, if require_all is true, */
  247. /* the result will be an /undefined error; if require_all is false, */
  248. /* the key will be ignored. */
  249. /* Note that .putdeviceparams clears the current pagedevice. */
  250. private int
  251. zputdeviceparams(os_ptr op)
  252. {    uint count = ref_stack_counttomark(&o_stack);
  253.     ref *prequire_all;
  254.     ref *ppolicy;
  255.     ref *pdev;
  256.     gx_device *dev;
  257.     stack_param_list list;
  258.     int code;
  259.     int old_width, old_height;
  260.     int i, dest;
  261.  
  262.     if ( count == 0 )
  263.       return_error(e_unmatchedmark);
  264.     prequire_all = ref_stack_index(&o_stack, count);
  265.     ppolicy = ref_stack_index(&o_stack, count + 1);
  266.     pdev = ref_stack_index(&o_stack, count + 2);
  267.     if ( pdev == 0 )
  268.       return_error(e_stackunderflow);
  269.     check_type_only(*prequire_all, t_boolean);
  270.     check_write_type_only(*pdev, t_device);
  271.     dev = pdev->value.pdevice;
  272.     code = stack_param_list_read(&list, &o_stack, 0, ppolicy,
  273.                      prequire_all->value.boolval);
  274.     if ( code < 0 )
  275.       return code;
  276.     old_width = dev->width;
  277.     old_height = dev->height;
  278.     code = gs_putdeviceparams(dev, (gs_param_list *)&list);
  279.     /* Check for names that were undefined or caused errors. */
  280.     for ( dest = count - 2, i = 0; i < count >> 1; i++ )
  281.       if ( list.results[i] < 0 )
  282.         {    *ref_stack_index(&o_stack, dest) =
  283.           *ref_stack_index(&o_stack, count - (i << 1) - 2);
  284.         gs_errorname(list.results[i],
  285.                  ref_stack_index(&o_stack, dest - 1));
  286.         dest -= 2;
  287.         }
  288.     iparam_list_release(&list);
  289.     if ( code < 0 )
  290.       {    /* There were errors reported. */
  291.         ref_stack_pop(&o_stack, dest + 1);
  292.         return 0;
  293.       }
  294.     if ( code > 0 || (code == 0 && (dev->width != old_width || dev->height != old_height)) )
  295.     {    /* The device was open and is now closed, */
  296.         /* or its dimensions have changed. */
  297.         /* If it was the current device, */
  298.         /* call setdevice to reinstall it and erase the page. */
  299.         /****** DOESN'T FIND ALL THE GSTATES THAT REFERENCE THE DEVICE. ******/
  300.         if ( gs_currentdevice(igs) == dev )
  301.         {    bool was_open = dev->is_open;
  302.             code = gs_setdevice_no_erase(igs, dev);
  303.             /* If the device wasn't closed, */
  304.             /* setdevice won't erase the page. */
  305.             if ( was_open && code >= 0 )
  306.               code = 1;
  307.         }
  308.     }
  309.     if ( code < 0 )
  310.       return code;
  311.     ref_stack_pop(&o_stack, count + 1);
  312.     make_bool(osp, code);
  313.     clear_pagedevice(istate);
  314.     return 0;
  315. }
  316.  
  317. /* <device> .setdevice <eraseflag> */
  318. /* Note that .setdevice clears the current pagedevice. */
  319. int
  320. zsetdevice(register os_ptr op)
  321. {    int code;
  322.  
  323.     check_write_type(*op, t_device);
  324.     code = gs_setdevice_no_erase(igs, op->value.pdevice);
  325.     if ( code < 0 )
  326.       return code;
  327.     make_bool(op, code != 0);    /* erase page if 1 */
  328.     clear_pagedevice(istate);
  329.     return code;
  330. }
  331.  
  332. /* ------ Initialization procedure ------ */
  333.  
  334. BEGIN_OP_DEFS(zdevice_op_defs) {
  335.     {"1copydevice", zcopydevice},
  336.     {"3copyscanlines", zcopyscanlines},
  337.     {"0currentdevice", zcurrentdevice},
  338.     {"1.devicename", zdevicename},
  339.     {"0.doneshowpage", zdoneshowpage},
  340.     {"0flushpage", zflushpage},
  341.     {"1.getdevice", zgetdevice},
  342.     {"2.getdeviceparams", zgetdeviceparams},
  343.     {"5makewordimagedevice", zmakewordimagedevice},
  344.     {"0nulldevice", znulldevice},
  345.     {"2.outputpage", zoutputpage},
  346.     {"3.putdeviceparams", zputdeviceparams},
  347.     {"1.setdevice", zsetdevice},
  348. END_OP_DEFS(0) }
  349.